/*
 * Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc.
 * Copyright (C) 2017 Furrtek
 *
 * This file is part of PortaPack.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2, or (at your option)
 * any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; see the file COPYING.  If not, write to
 * the Free Software Foundation, Inc., 51 Franklin Street,
 * Boston, MA 02110-1301, USA.
 */

#ifndef __SONDE_PACKET_H__
#define __SONDE_PACKET_H__

#include <cstdint>
#include <cstddef>

#include "field_reader.hpp"
#include "baseband_packet.hpp"
#include "manchester.hpp"

namespace sonde {

struct GPS_data {
    uint32_t alt{0};
    float lat{0};
    float lon{0};

    bool is_valid() const {
        if (lat >= -0.01 && lat <= 0.01 && lon >= -0.01 && lon <= 0.01)
            return false;
        if (lat < -90.0 || lat > 90.0)
            return false;
        if (lon < -180.0 || lon > 180.0)
            return false;
        return true;
    }
};

struct temp_humid {
    float temp{0};
    float humid{0};
};

class Packet {
   public:
    enum class Type : uint32_t {
        Unknown = 0,
        Meteomodem_unknown = 1,
        Meteomodem_M10 = 2,
        Meteomodem_M2K2 = 3,
        Vaisala_RS41_SG = 4,
        Meteomodem_M20 = 5,
    };

    Packet(const baseband::Packet& packet, const Type type);

    size_t length() const;

    Timestamp received_at() const;

    Type type() const;
    std::string type_string() const;

    std::string serial_number() const;
    uint32_t battery_voltage() const;
    GPS_data get_GPS_data() const;
    uint32_t frame() const;
    temp_humid get_temp_humid() const;
    float get_pressure() const;

    FormattedSymbols symbols_formatted() const;

    bool crc_ok() const;

   private:
    uint8_t getFwVerM20() const;
    static constexpr uint8_t vaisala_mask[64] = {
        0x96,
        0x83,
        0x3E,
        0x51,
        0xB1,
        0x49,
        0x08,
        0x98,
        0x32,
        0x05,
        0x59,
        0x0E,
        0xF9,
        0x44,
        0xC6,
        0x26,
        0x21,
        0x60,
        0xC2,
        0xEA,
        0x79,
        0x5D,
        0x6D,
        0xA1,
        0x54,
        0x69,
        0x47,
        0x0C,
        0xDC,
        0xE8,
        0x5C,
        0xF1,
        0xF7,
        0x76,
        0x82,
        0x7F,
        0x07,
        0x99,
        0xA2,
        0x2C,
        0x93,
        0x7C,
        0x30,
        0x63,
        0xF5,
        0x10,
        0x2E,
        0x61,
        0xD0,
        0xBC,
        0xB4,
        0xB6,
        0x06,
        0xAA,
        0xF4,
        0x23,
        0x78,
        0x6E,
        0x3B,
        0xAE,
        0xBF,
        0x7B,
        0x4C,
        0xC1};

    GPS_data ecef_to_gps() const;

    uint8_t vaisala_descramble(uint32_t pos) const;

    const baseband::Packet packet_;
    const BiphaseMDecoder decoder_;
    const FieldReader<BiphaseMDecoder, BitRemapNone> reader_bi_m;
    Type type_;

    using packetReader = FieldReader<baseband::Packet, BitRemapByteReverse>;  // baseband::Packet instead of BiphaseMDecoder

    bool crc_ok_M10() const;
    bool crc_ok_RS41() const;
    bool check_ok_M20() const;
    bool crc16rs41(uint32_t field_start) const;
};

} /* namespace sonde */

#endif /*__SONDE_PACKET_H__*/
